<!DOCTYPE html>
<html lang="zh-CN">
<head>
  <meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=2">
<meta name="theme-color" content="#222">
<meta name="generator" content="Hexo 4.2.1">
  <link rel="apple-touch-icon" sizes="180x180" href="/images/apple-touch-icon-next.png">
  <link rel="icon" type="image/png" sizes="32x32" href="/images/dute_favicon_32x32.png">
  <link rel="icon" type="image/png" sizes="16x16" href="/images/dute_favicon_16x16.png">
  <link rel="mask-icon" href="/images/logo.svg" color="#222">
  <link rel="manifest" href="/images/manifest.json">
  <meta name="msapplication-config" content="/images/browserconfig.xml">
  <meta http-equiv="Cache-Control" content="no-transform">
  <meta http-equiv="Cache-Control" content="no-siteapp">
  <meta name="google-site-verification" content="mpI5dkydstZXl6UcDCppqktXK0bbvqdZ6LkZ3KNk4Iw">
  <meta name="baidu-site-verification" content="code-a1LksZX2Ds">

<link rel="stylesheet" href="/css/main.css">


<link rel="stylesheet" href="/lib/font-awesome/css/font-awesome.min.css">
  <link rel="stylesheet" href="//cdn.jsdelivr.net/gh/fancyapps/fancybox@3/dist/jquery.fancybox.min.css">

<script id="hexo-configurations">
    var NexT = window.NexT || {};
    var CONFIG = {"hostname":"whitestore.top","root":"/","scheme":"Gemini","version":"7.8.0","exturl":true,"sidebar":{"position":"left","display":"post","padding":18,"offset":12,"onmobile":false},"copycode":{"enable":true,"show_result":false,"style":null},"back2top":{"enable":true,"sidebar":true,"scrollpercent":true},"bookmark":{"enable":false,"color":"#222","save":"auto"},"fancybox":true,"mediumzoom":false,"lazyload":false,"pangu":false,"comments":{"style":"tabs","active":null,"storage":true,"lazyload":false,"nav":null},"algolia":{"hits":{"per_page":10},"labels":{"input_placeholder":"Search for Posts","hits_empty":"We didn't find any results for the search: ${query}","hits_stats":"${hits} results found in ${time} ms"}},"localsearch":{"enable":true,"trigger":"auto","top_n_per_article":1,"unescape":false,"preload":false},"motion":{"enable":true,"async":false,"transition":{"post_block":"fadeIn","post_header":"slideDownIn","post_body":"slideDownIn","coll_header":"slideLeftIn","sidebar":"slideUpIn"}},"path":"search.xml"};
  </script>

  <meta name="description" content="类文件结构">
<meta property="og:type" content="article">
<meta property="og:title" content="深入理解JVM - 类文件结构">
<meta property="og:url" content="https://whitestore.top/2022/02/20/leiwenjian/index.html">
<meta property="og:site_name" content="爱看书的阿东">
<meta property="og:description" content="类文件结构">
<meta property="og:locale" content="zh_CN">
<meta property="og:image" content="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815195948.png">
<meta property="og:image" content="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210814183431.png">
<meta property="og:image" content="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815162054.png">
<meta property="og:image" content="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815165205.png">
<meta property="og:image" content="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815170559.png">
<meta property="og:image" content="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815183808.png">
<meta property="article:published_time" content="2022-02-20T12:52:32.000Z">
<meta property="article:modified_time" content="2023-07-16T06:28:09.272Z">
<meta property="article:author" content="阿东">
<meta property="article:tag" content="类文件结构">
<meta name="twitter:card" content="summary">
<meta name="twitter:image" content="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815195948.png">

<link rel="canonical" href="https://whitestore.top/2022/02/20/leiwenjian/">


<script id="page-configurations">
  // https://hexo.io/docs/variables.html
  CONFIG.page = {
    sidebar: "",
    isHome : false,
    isPost : true,
    lang   : 'zh-CN'
  };
</script>

  <title>深入理解JVM - 类文件结构 | 爱看书的阿东</title>
  






  <noscript>
  <style>
  .use-motion .brand,
  .use-motion .menu-item,
  .sidebar-inner,
  .use-motion .post-block,
  .use-motion .pagination,
  .use-motion .comments,
  .use-motion .post-header,
  .use-motion .post-body,
  .use-motion .collection-header { opacity: initial; }

  .use-motion .site-title,
  .use-motion .site-subtitle {
    opacity: initial;
    top: initial;
  }

  .use-motion .logo-line-before i { left: initial; }
  .use-motion .logo-line-after i { right: initial; }
  </style>
</noscript>

<link rel="alternate" href="/atom.xml" title="爱看书的阿东" type="application/atom+xml">
</head>

<body itemscope itemtype="http://schema.org/WebPage">
  <div class="container use-motion">
    <div class="headband"></div>

    <header class="header" itemscope itemtype="http://schema.org/WPHeader">
      <div class="header-inner"><div class="site-brand-container">
  <div class="site-nav-toggle">
    <div class="toggle" aria-label="切换导航栏">
      <span class="toggle-line toggle-line-first"></span>
      <span class="toggle-line toggle-line-middle"></span>
      <span class="toggle-line toggle-line-last"></span>
    </div>
  </div>

  <div class="site-meta">

    <a href="/" class="brand" rel="start">
      <span class="logo-line-before"><i></i></span>
      <h1 class="site-title">爱看书的阿东</h1>
      <span class="logo-line-after"><i></i></span>
    </a>
      <p class="site-subtitle" itemprop="description">赐他一块白色石头，石头上写着新名</p>
  </div>

  <div class="site-nav-right">
    <div class="toggle popup-trigger">
        <i class="fa fa-search fa-fw fa-lg"></i>
    </div>
  </div>
</div>




<nav class="site-nav">
  <ul id="menu" class="menu">
        <li class="menu-item menu-item-home">

    <a href="/" rel="section"><i class="fa fa-fw fa-home"></i>首页</a>

  </li>
        <li class="menu-item menu-item-tags">

    <a href="/tags/" rel="section"><i class="fa fa-fw fa-tags"></i>标签</a>

  </li>
        <li class="menu-item menu-item-categories">

    <a href="/categories/" rel="section"><i class="fa fa-fw fa-th"></i>分类</a>

  </li>
        <li class="menu-item menu-item-archives">

    <a href="/archives/" rel="section"><i class="fa fa-fw fa-archive"></i>归档</a>

  </li>
        <li class="menu-item menu-item-sitemap">

    <a href="/sitemap.xml" rel="section"><i class="fa fa-fw fa-sitemap"></i>站点地图</a>

  </li>
      <li class="menu-item menu-item-search">
        <a role="button" class="popup-trigger"><i class="fa fa-search fa-fw"></i>搜索
        </a>
      </li>
  </ul>
</nav>



  <div class="search-pop-overlay">
    <div class="popup search-popup">
        <div class="search-header">
  <span class="search-icon">
    <i class="fa fa-search"></i>
  </span>
  <div class="search-input-container">
    <input autocomplete="off" autocapitalize="off"
           placeholder="搜索..." spellcheck="false"
           type="search" class="search-input">
  </div>
  <span class="popup-btn-close">
    <i class="fa fa-times-circle"></i>
  </span>
</div>
<div id="search-result">
  <div id="no-result">
    <i class="fa fa-spinner fa-pulse fa-5x fa-fw"></i>
  </div>
</div>

    </div>
  </div>

</div>
    </header>

    

  <span class="exturl github-corner" data-url="aHR0cHM6Ly9naXRodWIuY29tL2xhenlUaW1lcw==" title="Follow me on GitHub" aria-label="Follow me on GitHub"><svg width="80" height="80" viewBox="0 0 250 250" aria-hidden="true"><path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path><path d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2" fill="currentColor" style="transform-origin: 130px 106px;" class="octo-arm"></path><path d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z" fill="currentColor" class="octo-body"></path></svg></span>


    <main class="main">
      <div class="main-inner">
        <div class="content-wrap">
          

          <div class="content post posts-expand">
            

    
  
  
  <article itemscope itemtype="http://schema.org/Article" class="post-block" lang="zh-CN">
    <link itemprop="mainEntityOfPage" href="https://whitestore.top/2022/02/20/leiwenjian/">

    <span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
      <meta itemprop="image" content="/images/avatar.gif">
      <meta itemprop="name" content="阿东">
      <meta itemprop="description" content="随遇而安">
    </span>

    <span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
      <meta itemprop="name" content="爱看书的阿东">
    </span>
      <header class="post-header">
        <h1 class="post-title" itemprop="name headline">
          深入理解JVM - 类文件结构
        </h1>

        <div class="post-meta">
            <span class="post-meta-item">
              <span class="post-meta-item-icon">
                <i class="fa fa-calendar-o"></i>
              </span>
              <span class="post-meta-item-text">发表于</span>

              <time title="创建时间：2022-02-20 20:52:32" itemprop="dateCreated datePublished" datetime="2022-02-20T20:52:32+08:00">2022-02-20</time>
            </span>
              <span class="post-meta-item">
                <span class="post-meta-item-icon">
                  <i class="fa fa-calendar-check-o"></i>
                </span>
                <span class="post-meta-item-text">更新于</span>
                <time title="修改时间：2023-07-16 14:28:09" itemprop="dateModified" datetime="2023-07-16T14:28:09+08:00">2023-07-16</time>
              </span>
            <span class="post-meta-item">
              <span class="post-meta-item-icon">
                <i class="fa fa-folder-o"></i>
              </span>
              <span class="post-meta-item-text">分类于</span>
                <span itemprop="about" itemscope itemtype="http://schema.org/Thing">
                  <a href="/categories/JVM/" itemprop="url" rel="index"><span itemprop="name">JVM</span></a>
                </span>
            </span>

          
            <span class="post-meta-item" title="阅读次数" id="busuanzi_container_page_pv" style="display: none;">
              <span class="post-meta-item-icon">
                <i class="fa fa-eye"></i>
              </span>
              <span class="post-meta-item-text">阅读次数：</span>
              <span id="busuanzi_value_page_pv"></span>
            </span>
  
  <span class="post-meta-item">
    
      <span class="post-meta-item-icon">
        <i class="fa fa-comment-o"></i>
      </span>
      <span class="post-meta-item-text">Valine：</span>
    
    <a title="valine" href="/2022/02/20/leiwenjian/#valine-comments" itemprop="discussionUrl">
      <span class="post-comments-count valine-comment-count" data-xid="/2022/02/20/leiwenjian/" itemprop="commentCount"></span>
    </a>
  </span>
  
  <br>
            <span class="post-meta-item" title="本文字数">
              <span class="post-meta-item-icon">
                <i class="fa fa-file-word-o"></i>
              </span>
                <span class="post-meta-item-text">本文字数：</span>
              <span>5.1k</span>
            </span>
            <span class="post-meta-item" title="阅读时长">
              <span class="post-meta-item-icon">
                <i class="fa fa-clock-o"></i>
              </span>
                <span class="post-meta-item-text">阅读时长 &asymp;</span>
              <span>5 分钟</span>
            </span>
            <div class="post-description">类文件结构</div>

        </div>
      </header>

    
    
    
    <div class="post-body" itemprop="articleBody">

      
        <h1 id="深入理解JVM-类文件结构"><a href="#深入理解JVM-类文件结构" class="headerlink" title="深入理解JVM - 类文件结构"></a>深入理解JVM - 类文件结构</h1><h1 id="前言"><a href="#前言" class="headerlink" title="前言"></a>前言</h1><p>​    JVM的类文件结构基本都会要记忆的内容，我相信你也记不住，当然我也是记不住的，所以这里只会列出大致的类文件结构，我们需要大致了解类文件结构是怎么一回事就行了，具体到那个位存哪个内容，内容确实太多了，感兴趣可以直接去读书中对应的<code>第6章　类文件结构</code>这一个章节的内容。</p>
<p>​    类文件结构个人认为需要注意的点就是这几点：大致的类文件结构，部分Jdk的特性如何通过改动class文件结构实现，比如泛型，自动拆装箱，动态代理，lambada语法等。</p>
<h1 id="概述："><a href="#概述：" class="headerlink" title="概述："></a>概述：</h1><p>​    其实主要内容就是介绍CLASS的文件结构。</p>
<ol>
<li>了解JVM的类文件基本结构。</li>
<li>了解常量池的内容</li>
<li>了解重点内容属性表集合</li>
</ol>
<a id="more"></a>

<h1 id="思维导图"><a href="#思维导图" class="headerlink" title="思维导图"></a>思维导图</h1><p>​    下面是思维导图的地址：<span class="exturl" data-url="aHR0cHM6Ly93d3cubXVidWNtLmNvbS9kb2MvMS1aR2MtMDdrdkI=" title="https://www.mubucm.com/doc/1-ZGc-07kvB">https://www.mubucm.com/doc/1-ZGc-07kvB<i class="fa fa-external-link"></i></span></p>
<p><img src="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815195948.png" alt=""></p>
<h1 id="什么是Class类文件？"><a href="#什么是Class类文件？" class="headerlink" title="什么是Class类文件？"></a>什么是Class类文件？</h1><p>​    <code>.class</code>文件是由<code>.java</code>通过<code>Javac</code>的命令编译而来的，也是JVM实现跨平台的关键，同时Class类文件实际上的内容是包含<strong>字节码指令</strong>的二进制文件，而字节码指令简单理解是jvm对于汇编指令的进一步封装，甚至有一些书籍拿字节码的指令来讲部分操作系统的底层逻辑实现，注意不要被洗脑了，JVM的字节码指令只能被JVM识别，放到别的平台就是一堆乱码，如果带歪了建议看CSAPP这本书洗回来。</p>
<p>​    既然是由外部的.java文件翻译并且加载到虚拟机上，那么class文件的结构毫无疑问需要严格的规定，防止代码破坏jvm正常执行，事实上《JVM虚拟机规范》规定了Class文件的整个结构，对于每一位都有严格的要求，现代的JDK虽然对于这个规范有了不少的改动，但是整体来看最基础的结构还是按照最初始发布的那一套执行，所以基本不需要担心过时的问题。</p>
<p>​    <strong>任意一个class文件对应一个类或者接口定义</strong>，class文件结构也可以看做是一种规范，只要其他语言也能遵守class文件的规范，意味着完全可以通过编写其他语言的程序转化为class文件最终翻译到JVM中。</p>
<h1 id="class文件结构"><a href="#class文件结构" class="headerlink" title="class文件结构"></a>class文件结构</h1><p>​    Class文件是一组<strong>以8个字节为基础</strong>单位的<strong>二进制流</strong>，各个数据项目<strong>严格按照顺序</strong>紧凑地排列在文件之中，中间<strong>没有添加任何分隔符</strong>，这使得整个Class文件中存储的内容几乎全部是程序运行的必要数<br>据，没有空隙存在。当遇到需要占用8个字节以上空间的数据项时，则会按照高位在前 [2] 的方式分割<br>成若干个8个字节进行存储。</p>
<p>​    Class文件格式采用一种<strong>类似于C语言结构体</strong>的伪结构来存储数据，这种伪结构中只有两种数据类型：<strong>“无符号数”和“表”</strong>。</p>
<p>​    <strong>无符号数</strong>：代表了基本的数据类型，比如u1、u2、u3等，数字代表了字节，比如1个字节，2个字节，3个字节，无符号数可以描述数字，索引引用或者经过UTF-8的编码为字符串存储</p>
<blockquote>
<p>比如<code>\u304</code>这种字符串</p>
</blockquote>
<p>​    <strong>表</strong>：表是由<strong>多个无符号数</strong>或者其他表作为数据项构成的复合数据类型，习惯性以“<strong>_info</strong>”结尾。</p>
<p>​    最后可以整个class文件结构看出如下的一张表。</p>
<p>​    <img src="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210814183431.png" alt=""></p>
<h1 id="class文件结构详解"><a href="#class文件结构详解" class="headerlink" title="class文件结构详解"></a>class文件结构详解</h1><p>​    了解了class文件的大致结构，下面来聊聊class文件的具体组成了。</p>
<h2 id="魔数0xCAFEBABE"><a href="#魔数0xCAFEBABE" class="headerlink" title="魔数0xCAFEBABE"></a>魔数<code>0xCAFEBABE</code></h2><p>​    在class文件的结构中，每个Class文件的头4个字节被称为魔数（Magic Number），它唯一的作用是标记这个文件是一个class文件，除此之外没有其他作用，至于为什么叫做咖啡宝贝是因为它象征着著名咖啡品牌Peet’s Coffee并且深受欢迎的Baristas咖啡。</p>
<h2 id="次主版本号"><a href="#次主版本号" class="headerlink" title="次主版本号"></a>次主版本号</h2><p>​    为什么叫做次主版本号？是因为接着魔数的后面的位数第5、6个字节被称为次版本号，第7、8个字节是主版本号。Java的版本号是从45开始的，JDK 1.1之后的每个JDK大版本发布主版本号向上加1（JDK 1.0～1.1使用了45.0～45.3的版本号）。<strong>高版本支持向下兼容，但是低版本不支持向上兼容，哪怕代码一模一样。</strong>《Java虚拟机规范》</p>
<blockquote>
<p>​    例如，JDK 1.1能支持版本号为45.0～45.65535的Class文件，无法执行版本号为46.0以上的Class文件，而JDK 1.2则能支持45.0～46.65535的Class文件。目前最新的JDK版本为13，可生成的Class文件主版本号最大值为57.0。</p>
</blockquote>
<p>​    下面是书中给出的具体案例：</p>
<p><img src="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815162054.png" alt=""></p>
<p>下面是一个JDK版本号对应参考图：</p>
<p><img src="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815165205.png" alt=""></p>
<blockquote>
<p>从JDK 9开始，Javac编译器不再支持使用-source参数编译版本号小于1.5的源码</p>
<p>原因：JDK9的<strong>模块化</strong>，以及<strong>扩展类加载器</strong>的改动。</p>
</blockquote>
<h2 id="常量池"><a href="#常量池" class="headerlink" title="常量池"></a>常量池</h2><blockquote>
<p>​    注意：常量池的入口需要放置一个U2（16进制的F）类型的数据，用于记录<strong>常量池的容器计量值</strong></p>
</blockquote>
<p>​    主次版本号之后就是常量池的内容了，可以直接看做是class文件的资源仓库，同时是class文件结构关联最多的数据部分，也是最大的数据项之一。光靠这一篇文章肯定是无法讲完的，同样即使讲完了也记不住，所以这一部分我们只需要掌握存放的内容即可。</p>
<h3 id="存放内容"><a href="#存放内容" class="headerlink" title="存放内容"></a>存放内容</h3><p>​    常量池中主要存放两大类常量：<strong>字面量（Literal）</strong>和<strong>符号引用（Symbolic References）</strong>。</p>
<p>​    字面量：比如文本字符串，final常量</p>
<p>​    符号引用则存放如下内容：</p>
<ul>
<li>模块导出或者开放包</li>
<li>类与接口全限定名称</li>
<li>字段与描述符</li>
<li>方法句柄和类型</li>
<li>动态调用点和动态常量</li>
</ul>
<h3 id="常量表"><a href="#常量表" class="headerlink" title="常量表"></a>常量表</h3><p>​    常量池的每一个常量都是一个表，最初只有11种结构，后来扩展出<strong>4种</strong>和动态语言相关的常量，为了模块化：加入CONSTANT_Module_info和CONSTANT_Package_info两个常量，最终就是11+4+2 = 17种常量，所以到JDK16版本为止有17种常量。同意，记是记不住的，我们也不需要记住：</p>
<p><img src="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815170559.png" alt=""></p>
<blockquote>
<p>上面的表重点关注:<strong>CONSTANT_Dynamic_info</strong>和<strong>CONSTANT_InvokeDynamic_info</strong></p>
</blockquote>
<p>​    下面我们挑重点看一下这些常量对于JDK的影响。</p>
<h4 id="Constant-class-info-类型"><a href="#Constant-class-info-类型" class="headerlink" title="Constant_class_info 类型"></a>Constant_class_info 类型</h4><p>​    <strong>CONSTANT_Class_info</strong>类型，此类型的常量<strong>代表一个类或者接口的符号引用</strong>，主要结构为一个U1类型的tag（标志位）和u2类型的name_index（名称引用）。</p>
<p>​    tag的作用是标志位区分常量的类型，而name_index表示常量池的索引值。</p>
<blockquote>
<p>U1代表一个字节：1111，U2代表两个字节：11111111，也就是65535。</p>
<p>后续以此类推，不再赘述。</p>
</blockquote>
<p>​    在讲下一个类型之前，我们先提一个问题：<strong>JAVA方法最大长度是多少，为什么？</strong>我们都知道方法起名是有上限的，但是究竟的上限是多少，这里我们根据class的文件结构来进行解读：</p>
<h4 id="CONSTANT-Utf8-info类型"><a href="#CONSTANT-Utf8-info类型" class="headerlink" title="CONSTANT_Utf8_info类型"></a>CONSTANT_Utf8_info类型</h4><p>​    <strong>CONSTANT_Utf8_info</strong>类型常量，此常量代表了这个<strong>类（或者接口）的全限定名</strong>。</p>
<p>​    <strong>CONSTANT_Utf8_info</strong>类型常量指向name_index，0x00002常量池第二项常量，标记为<code>0x01</code></p>
<p>。接下来就是重点了，<strong>CONSTANT_Utf8_info</strong>型常量的最大长度也就是<strong>Java中方法、字段名的最大长度</strong>。而这里的最大长度就是length的最大值，既u2类型能表达的最大值65535。所以Java程序中如果定义了超过64KB英文字符的变量或方法名，即使规则和全部字符都是合法的，也会<strong>无法编译</strong>。</p>
<p>​    书中提到的常量也是这两种，如果我们想要查看字节码，可以使用<code>javap</code>指令。</p>
<blockquote>
<p>JDK 7时增加了前三种：CONSTANT_MethodHandle_info、CONSTANT_MethodType_info和<br>CONSTANT_InvokeDynamic_info。DK 11中又增加了第四种常量CONSTANT_Dynamic_info</p>
</blockquote>
<h2 id="访问标志"><a href="#访问标志" class="headerlink" title="访问标志"></a>访问标志</h2><p>​    在常量池结束之后，用两个字节表示访问标志与接口的访问信息。里面的具体内容包括：这个Class是类还是接口；是否定义为public类型；是否定义为abstract类型；如果是类的话，是否被声明为final</p>
<p>​    access_flags（访问标志）中一共有16个标志位可以使用，当前只定义了其中9个，如果要计算它的值，可以使用<code>0x0001|0x0020=0x0021</code>。</p>
<p><img src="https://gitee.com/lazyTimes/imageReposity/raw/master/img/20210815183808.png" alt=""></p>
<p>​    </p>
<h2 id="类索引、父类索引与接口索引集合"><a href="#类索引、父类索引与接口索引集合" class="headerlink" title="类索引、父类索引与接口索引集合"></a>类索引、父类索引与接口索引集合</h2><p>​    简单来说，Class文件中由这三项数据来确定<strong>该类型的继承关系</strong>。类索引用于确定<strong>这个类的全限定名</strong>，而父类索引中记录了当前类的父类的全限定名，需要注意的是因为JAVA的顶级父类永远是java.lang.Object，除了java.lang.Object外，所有Java类的父类索引都不为0，而接口索引集合就用来描述这个类实现了哪些接口。</p>
<h2 id="字段表集合"><a href="#字段表集合" class="headerlink" title="字段表集合"></a>字段表集合</h2><p>​    字段表（field_info）用于描述接口或者类中声明的变量，注意字段包含了java当中的类变量或者实例级常量。这里可能有一个疑问，<strong>为什么方法中的局部变量不属于字段</strong>？**。</p>
<p>​    首先，我们需要注意的是，方法中的变量都是属于栈帧范围内的，所以方法中的变量生命范围逃不开一个栈帧的高度（或者说容量）。类中的字段可以定义是否公开，是否静态，引用是否不可修改等，但是局部变量做不到，所以字段表中无法存放局部变量，也没有必要存放。</p>
<p>​    同样的，上述这些信息中，各个修饰符都是布尔值，要么有某个修饰符，要么没有，很适合使用标志位来表示。</p>
<p>​    接下来书里面的内容就是讲述字段表集合的细节了，由于我们不需要去研究GC，所以这里感兴趣可以直接去看书里面的内容。</p>
<h2 id="方法表集合"><a href="#方法表集合" class="headerlink" title="方法表集合"></a>方法表集合</h2><p>​    和字段表内容大致相同，Class文件存储格式中对方法的描述与对字段的描述采用了几乎完全一致的方式，方法表集合包括了：<strong>访问标志（access_flags）、名称索引（name_index）、描述符索引（descriptor_index）、属性表集合（attributes）</strong>。同时在标志部分增加了和方法相关的标志。</p>
<p>​    既然是方法表集合，那代码到哪去了？答案是在方法属性里面有一个code属性，这属性就是存放方法中代码的地方，也是对于程序员来说最有用的地方。这里需要注意需要注意的是如果父类方法在子类中没有被重写（Override），方法表集合中就<strong>不会出现来自父类的方法信息</strong>。</p>
<p>​    同样，需要注意的是虽然java语法不支持返回值重载但是在class特征签名返回值不同也可以同时存在。</p>
<blockquote>
<p>需要注意的是volatile关键字和transient关键字在方法表集合当中不一样，</p>
</blockquote>
<h2 id="属性表集合-核心"><a href="#属性表集合-核心" class="headerlink" title="属性表集合(核心)"></a>属性表集合(核心)</h2><p>​    属性表的内容主要是搭配前面所讲的字段和方法进行搭配的，最初的预定义属性最初只有9种，最新的《Java虚拟机规范》的Java SE 12版本中，现在已经有了29种。</p>
<figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line">属性结构的内容如下：</span><br><span class="line">- u2 attrcibute_name_index</span><br><span class="line">- U4 attribute_length</span><br><span class="line">- U1 info</span><br></pre></td></tr></table></figure>



<h3 id="Code属性"><a href="#Code属性" class="headerlink" title="Code属性"></a>Code属性</h3><p>​    Java程序方法体里面的代码经过Javac编译器处理之后，最终变为字节码指令存储在Code属性内。但是需要注意并不是所有的方法都要有这个属性，因为方法的内容是<strong>可以为空</strong>的。</p>
<p>​    Code属性是Class文件中最重要的一个属性，如果把一个Java程序中的信息分为代码（Code，方法<br>体里面的Java代码）和元数据（Metadata，包括类、字段、方法定义及其他信息）两部分，那么在整<br>个Class文件里，<strong>Code属性用于描述代码，所有的其他数据项目都用于描述元数据</strong>。</p>
<p>​    后续的内容是根据的一个javap生产的字节码指令来进行相关的解读，这些内容也是不是关键的内容，为了减轻记忆负担，本文也不做过多介绍。</p>
<h3 id="异常表"><a href="#异常表" class="headerlink" title="异常表"></a>异常表</h3><p>​    Code数量里面还包含一个异常表，异常表也是JAVA代码的一部分，这部分内容虽然可以通过GOTO这种跳转指令实现，但是在JVM规范中是强制规范JAVA语言使用异常表而不是GOTO指令实现JAVA的异常以及Finally的处理机制。</p>
<blockquote>
<p>为什么不能用GOTO？这就要问问C语言这个老先生了，虽然很多语言都保留了GOTO的语法，但是无一例外没有人推荐使用。因为它不仅容易出BUG，并且写出来的源代码十分难以理解。</p>
</blockquote>
<h3 id="line-number-table-属性"><a href="#line-number-table-属性" class="headerlink" title="line_number_table 属性"></a>line_number_table 属性</h3><ul>
<li><p>用于描述行号和字节码行号对应关系</p>
</li>
<li><p>用于debug使用</p>
</li>
<li><p><strong>-line_number_info</strong> 中两个u2类型的数据项</p>
<ul>
<li>start_pc: 字节码行号</li>
<li>line_number java源代码行号</li>
</ul>
</li>
</ul>
<p>由于篇幅问题，其他的属性这里就不过多介绍了，思维导图摘录了大致内容，在遇到疑问的时候翻一翻记忆会比较深，这里也不做过多叙述。</p>
<h1 id="总结"><a href="#总结" class="headerlink" title="总结"></a>总结</h1><p>​    class文件结构靠着死记硬背是记不住的，需要根据JVM的特性来进行理解，比如动态语言是如何实现的，再比如DEBUG是如何实现的等等，用这些内容帮助理解记忆。</p>
<p>​    从本文也可以看到，其实重点都在<strong>属性表集合</strong>这一部分。所以如果需要重点理解JAVA的特性，可以从这个属性表开始。</p>
<h1 id="写在最后"><a href="#写在最后" class="headerlink" title="写在最后"></a>写在最后</h1><p>​    不管看几次，我也记不住，哎……</p>
<script type="text/javascript" src="https://cdn.jsdelivr.net/npm/kity@2.0.4/dist/kity.min.js"></script><script type="text/javascript" src="https://cdn.jsdelivr.net/npm/kityminder-core@1.4.50/dist/kityminder.core.min.js"></script><script defer="true" type="text/javascript" src="https://cdn.jsdelivr.net/npm/hexo-simple-mindmap@0.2.0/dist/mindmap.min.js"></script><link rel="stylesheet" type="text/css" href="https://cdn.jsdelivr.net/npm/hexo-simple-mindmap@0.2.0/dist/mindmap.min.css">
    </div>

    
    
    
        

<div>
<ul class="post-copyright">
  <li class="post-copyright-author">
    <strong>本文作者： </strong>阿东
  </li>
  <li class="post-copyright-link">
    <strong>本文链接：</strong>
    <a href="https://whitestore.top/2022/02/20/leiwenjian/" title="深入理解JVM - 类文件结构">https://whitestore.top/2022/02/20/leiwenjian/</a>
  </li>
  <li class="post-copyright-license">
    <strong>版权声明： </strong>本博客所有文章除特别声明外，均采用 <span class="exturl" data-url="aHR0cHM6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL2xpY2Vuc2VzL2J5LW5jLzQuMC96aC1DTg=="><i class="fa fa-fw fa-creative-commons"></i>BY-NC</span> 许可协议。转载请注明出处！
  </li>
</ul>
</div>


      <footer class="post-footer">
          <div class="post-tags">
              <a href="/tags/JVM/" rel="tag"># JVM</a>
          </div>

        


        
    <div class="post-nav">
      <div class="post-nav-item">
    <a href="/2022/02/20/jvmzijiema/" rel="prev" title="深入理解JVM - 字节码指令">
      <i class="fa fa-chevron-left"></i> 深入理解JVM - 字节码指令
    </a></div>
      <div class="post-nav-item">
    <a href="/2022/02/20/dongtaileixing/" rel="next" title="深入理解JVM - 动态类型语言">
      深入理解JVM - 动态类型语言 <i class="fa fa-chevron-right"></i>
    </a></div>
    </div>
      </footer>
    
  </article>
  
  
  



          </div>
          
    <div class="comments" id="valine-comments"></div>

<script>
  window.addEventListener('tabs:register', () => {
    let { activeClass } = CONFIG.comments;
    if (CONFIG.comments.storage) {
      activeClass = localStorage.getItem('comments_active') || activeClass;
    }
    if (activeClass) {
      let activeTab = document.querySelector(`a[href="#comment-${activeClass}"]`);
      if (activeTab) {
        activeTab.click();
      }
    }
  });
  if (CONFIG.comments.storage) {
    window.addEventListener('tabs:click', event => {
      if (!event.target.matches('.tabs-comment .tab-content .tab-pane')) return;
      let commentClass = event.target.classList[1];
      localStorage.setItem('comments_active', commentClass);
    });
  }
</script>

        </div>
          
  
  <div class="toggle sidebar-toggle">
    <span class="toggle-line toggle-line-first"></span>
    <span class="toggle-line toggle-line-middle"></span>
    <span class="toggle-line toggle-line-last"></span>
  </div>

  <aside class="sidebar">
    <div class="sidebar-inner">

      <ul class="sidebar-nav motion-element">
        <li class="sidebar-nav-toc">
          文章目录
        </li>
        <li class="sidebar-nav-overview">
          站点概览
        </li>
      </ul>

      <!--noindex-->
      <div class="post-toc-wrap sidebar-panel">
          <div class="post-toc motion-element"><ol class="nav"><li class="nav-item nav-level-1"><a class="nav-link" href="#深入理解JVM-类文件结构"><span class="nav-number">1.</span> <span class="nav-text">深入理解JVM - 类文件结构</span></a></li><li class="nav-item nav-level-1"><a class="nav-link" href="#前言"><span class="nav-number">2.</span> <span class="nav-text">前言</span></a></li><li class="nav-item nav-level-1"><a class="nav-link" href="#概述："><span class="nav-number">3.</span> <span class="nav-text">概述：</span></a></li><li class="nav-item nav-level-1"><a class="nav-link" href="#思维导图"><span class="nav-number">4.</span> <span class="nav-text">思维导图</span></a></li><li class="nav-item nav-level-1"><a class="nav-link" href="#什么是Class类文件？"><span class="nav-number">5.</span> <span class="nav-text">什么是Class类文件？</span></a></li><li class="nav-item nav-level-1"><a class="nav-link" href="#class文件结构"><span class="nav-number">6.</span> <span class="nav-text">class文件结构</span></a></li><li class="nav-item nav-level-1"><a class="nav-link" href="#class文件结构详解"><span class="nav-number">7.</span> <span class="nav-text">class文件结构详解</span></a><ol class="nav-child"><li class="nav-item nav-level-2"><a class="nav-link" href="#魔数0xCAFEBABE"><span class="nav-number">7.1.</span> <span class="nav-text">魔数0xCAFEBABE</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#次主版本号"><span class="nav-number">7.2.</span> <span class="nav-text">次主版本号</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#常量池"><span class="nav-number">7.3.</span> <span class="nav-text">常量池</span></a><ol class="nav-child"><li class="nav-item nav-level-3"><a class="nav-link" href="#存放内容"><span class="nav-number">7.3.1.</span> <span class="nav-text">存放内容</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#常量表"><span class="nav-number">7.3.2.</span> <span class="nav-text">常量表</span></a><ol class="nav-child"><li class="nav-item nav-level-4"><a class="nav-link" href="#Constant-class-info-类型"><span class="nav-number">7.3.2.1.</span> <span class="nav-text">Constant_class_info 类型</span></a></li><li class="nav-item nav-level-4"><a class="nav-link" href="#CONSTANT-Utf8-info类型"><span class="nav-number">7.3.2.2.</span> <span class="nav-text">CONSTANT_Utf8_info类型</span></a></li></ol></li></ol></li><li class="nav-item nav-level-2"><a class="nav-link" href="#访问标志"><span class="nav-number">7.4.</span> <span class="nav-text">访问标志</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#类索引、父类索引与接口索引集合"><span class="nav-number">7.5.</span> <span class="nav-text">类索引、父类索引与接口索引集合</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#字段表集合"><span class="nav-number">7.6.</span> <span class="nav-text">字段表集合</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#方法表集合"><span class="nav-number">7.7.</span> <span class="nav-text">方法表集合</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#属性表集合-核心"><span class="nav-number">7.8.</span> <span class="nav-text">属性表集合(核心)</span></a><ol class="nav-child"><li class="nav-item nav-level-3"><a class="nav-link" href="#Code属性"><span class="nav-number">7.8.1.</span> <span class="nav-text">Code属性</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#异常表"><span class="nav-number">7.8.2.</span> <span class="nav-text">异常表</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#line-number-table-属性"><span class="nav-number">7.8.3.</span> <span class="nav-text">line_number_table 属性</span></a></li></ol></li></ol></li><li class="nav-item nav-level-1"><a class="nav-link" href="#总结"><span class="nav-number">8.</span> <span class="nav-text">总结</span></a></li><li class="nav-item nav-level-1"><a class="nav-link" href="#写在最后"><span class="nav-number">9.</span> <span class="nav-text">写在最后</span></a></li></ol></div>
      </div>
      <!--/noindex-->

      <div class="site-overview-wrap sidebar-panel">
        <div class="site-author motion-element" itemprop="author" itemscope itemtype="http://schema.org/Person">
  <p class="site-author-name" itemprop="name">阿东</p>
  <div class="site-description" itemprop="description">随遇而安</div>
</div>
<div class="site-state-wrap motion-element">
  <nav class="site-state">
      <div class="site-state-item site-state-posts">
          <a href="/archives/">
        
          <span class="site-state-item-count">239</span>
          <span class="site-state-item-name">日志</span>
        </a>
      </div>
      <div class="site-state-item site-state-categories">
            <a href="/categories/">
          
        <span class="site-state-item-count">36</span>
        <span class="site-state-item-name">分类</span></a>
      </div>
      <div class="site-state-item site-state-tags">
            <a href="/tags/">
          
        <span class="site-state-item-count">37</span>
        <span class="site-state-item-name">标签</span></a>
      </div>
  </nav>
</div>
  <div class="links-of-author motion-element">
      <span class="links-of-author-item">
        <span class="exturl" data-url="aHR0cHM6Ly9naXRodWIuY29tL2xhenlUaW1lcw==" title="GitHub → https:&#x2F;&#x2F;github.com&#x2F;lazyTimes"><i class="fa fa-fw fa-github"></i>GitHub</span>
      </span>
      <span class="links-of-author-item">
        <span class="exturl" data-url="bWFpbHRvOjEwOTc0ODM1MDhAcXEuY29t" title="E-Mail → mailto:1097483508@qq.com"><i class="fa fa-fw fa-envelope"></i>E-Mail</span>
      </span>
  </div>


  <div class="links-of-blogroll motion-element">
    <div class="links-of-blogroll-title">
      <i class="fa fa-fw fa-link"></i>
      友情链接
    </div>
    <ul class="links-of-blogroll-list">
        <li class="links-of-blogroll-item">
          <span class="exturl" data-url="aHR0cHM6Ly93d3cuNTJwb2ppZS5jbi9ob21lLnBocD9tb2Q9c3BhY2UmdWlkPTE0OTc3MTgmZG89dGhyZWFkJnZpZXc9bWUmZnJvbT1zcGFjZQ==" title="https:&#x2F;&#x2F;www.52pojie.cn&#x2F;home.php?mod&#x3D;space&amp;uid&#x3D;1497718&amp;do&#x3D;thread&amp;view&#x3D;me&amp;from&#x3D;space">吾爱破解</span>
        </li>
        <li class="links-of-blogroll-item">
          <span class="exturl" data-url="aHR0cHM6Ly9qdWVqaW4uaW0vdXNlci8yOTk5MTIzNDUyNjI2MzY2" title="https:&#x2F;&#x2F;juejin.im&#x2F;user&#x2F;2999123452626366">掘金</span>
        </li>
        <li class="links-of-blogroll-item">
          <span class="exturl" data-url="aHR0cHM6Ly9zZWdtZW50ZmF1bHQuY29tL3UvbGF6eXRpbWVz" title="https:&#x2F;&#x2F;segmentfault.com&#x2F;u&#x2F;lazytimes">思否</span>
        </li>
    </ul>
  </div>

      </div>

      <div class="wechat_OA">
        <span>欢迎关注我的公众号</span>
        <br>
          <!-- 这里添加你的二维码图片 -->
        <img src ="https://adong-picture.oss-cn-shenzhen.aliyuncs.com/adong/wechat_channel.jpg">
      </div>
        <div class="back-to-top motion-element">
          <i class="fa fa-arrow-up"></i>
          <span>0%</span>
        </div>

    </div>
  </aside>
  <div id="sidebar-dimmer"></div>


      </div>
    </main>

    <footer class="footer">
      <div class="footer-inner">
        

        

<div class="copyright">
  
  &copy; 
  <span itemprop="copyrightYear">2023</span>
  <span class="with-love">
    <i class="fa fa-user"></i>
  </span>
  <span class="author" itemprop="copyrightHolder">阿东</span>
    <span class="post-meta-divider">|</span>
    <span class="post-meta-item-icon">
      <i class="fa fa-area-chart"></i>
    </span>
      <span class="post-meta-item-text">站点总字数：</span>
    <span title="站点总字数">2m</span>
    <span class="post-meta-divider">|</span>
    <span class="post-meta-item-icon">
      <i class="fa fa-coffee"></i>
    </span>
      <span class="post-meta-item-text">站点阅读时长 &asymp;</span>
    <span title="站点阅读时长">29:50</span>
</div>
  <div class="powered-by">由 <span class="exturl theme-link" data-url="aHR0cHM6Ly9oZXhvLmlv">Hexo</span> & <span class="exturl theme-link" data-url="aHR0cHM6Ly90aGVtZS1uZXh0Lm9yZw==">NexT.Gemini</span> 强力驱动
  </div>

        
<div class="busuanzi-count">
  <script async src="https://busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js"></script>
    <span class="post-meta-item" id="busuanzi_container_site_uv" style="display: none;">
      <span class="post-meta-item-icon">
        <i class="fa fa-user"></i>
      </span>
      <span class="site-uv" title="总访客量">
        <span id="busuanzi_value_site_uv"></span>
      </span>
    </span>
    <span class="post-meta-divider">|</span>
    <span class="post-meta-item" id="busuanzi_container_site_pv" style="display: none;">
      <span class="post-meta-item-icon">
        <i class="fa fa-eye"></i>
      </span>
      <span class="site-pv" title="总访问量">
        <span id="busuanzi_value_site_pv"></span>
      </span>
    </span>
</div>








      </div>
    </footer>
  </div>

  
  <script src="/lib/anime.min.js"></script>
  <script src="//cdn.jsdelivr.net/npm/jquery@3/dist/jquery.min.js"></script>
  <script src="//cdn.jsdelivr.net/gh/fancyapps/fancybox@3/dist/jquery.fancybox.min.js"></script>
  <script src="/lib/velocity/velocity.min.js"></script>
  <script src="/lib/velocity/velocity.ui.min.js"></script>

<script src="/js/utils.js"></script>

<script src="/js/motion.js"></script>


<script src="/js/schemes/pisces.js"></script>


<script src="/js/next-boot.js"></script>




  




  
<script src="/js/local-search.js"></script>













  

  


<script>
NexT.utils.loadComments(document.querySelector('#valine-comments'), () => {
  NexT.utils.getScript('//unpkg.com/valine/dist/Valine.min.js', () => {
    var GUEST = ['nick', 'mail', 'link'];
    var guest = 'nick,mail,link';
    guest = guest.split(',').filter(item => {
      return GUEST.includes(item);
    });
    new Valine({
      el         : '#valine-comments',
      verify     : false,
      notify     : true,
      appId      : 'qMUpEEvBgXaMDD1b0ftgi9xr-gzGzoHsz',
      appKey     : 'UCdfT4Rfih6MO6y8DI4fstf6',
      placeholder: "Just go go",
      avatar     : 'mm',
      meta       : guest,
      pageSize   : '10' || 10,
      visitor    : false,
      lang       : 'zh-CN' || 'zh-cn',
      path       : location.pathname,
      recordIP   : false,
      serverURLs : ''
    });
  }, window.Valine);
});
</script>

</body>
</html>
